home *** CD-ROM | disk | FTP | other *** search
/ TeX 1995 July / TeX CD-ROM July 1995 (Disc 1)(Walnut Creek)(1995).ISO / dviware / ln03 / thomas / ln03dvi.c < prev    next >
C/C++ Source or Header  |  1990-10-01  |  54KB  |  1,933 lines

  1. /* Ln03DVI translates a TeX DVI file to an LN03 format file. 
  2.  
  3. Ln03DVI is still being developed. Copyright (c) 1985, 1986, 1987
  4. by Digital Equipment Corporation, Maynard, Massachusetts,
  5. USA. Originial Author: Flavio Rose (...!decwrl!dvinci.dec.com!rose)
  6.  
  7. This version of Ln03DVI is maintained by Matt Thomas.
  8. Please send all bug reports to either:
  9.     ...!decwrl!thebay.dec.com!mthomas (UUCP)
  10.     mthomas@thebay.dec.com (Internet) */
  11.  
  12. /* Dvi2ln3 is based on the publicly-available program DVItype,
  13. written by David R. Fuchs of Stanford University; and also
  14. on earlier DEC programs, Dvi2lng, Topp and LN03Topp, for the
  15. LN01 and LN03 laser printers. This program is not a DEC
  16. product and is not guaranteed to work. 
  17.  
  18. Dvi2ln3 is written in VAX C. Specific VAX and VMS
  19. dependencies have generally been avoided, however. The
  20. reader may find them by searching for the strings VAX and
  21. VMS in this file. 
  22.  
  23. During development, double square brackets [[ ]] in a
  24. comment indicate some places where the code needs to be
  25. improved. 
  26.  
  27. [[Among the useful things that still need to be done:
  28. differentiating between int and long variables so this can
  29. be ported more easily to 16-bit architectures; testing to
  30. see if malloc returns 0; a \special for ROM fonts.]] 
  31.  
  32. Development history:
  33.  
  34. Feb. 85:    Did some early work on Dvi2ln3, translating bits
  35.         and pieces of Dvi2lng. Concluded, however, that it
  36.         was better to translate LN03Topp and Dvi2lng into
  37.         C first. The reason for this is that code which
  38.         mimics an existing program is easier to write and
  39.         to test. 
  40.  
  41. 5/15/85:    At this date, the old LN03Topp program has been
  42.         successfully translated into C (except for the
  43.         ability to read files over DECnet). At this point,
  44.         did some further cleanup of the Dvi2ln3 source. 
  45.  
  46. 6/4/85:        Coding of Dvi2ln3 begins in earnest. Merging bits
  47.         and pieces from various places. Chucking chunks
  48.         and all that junk from LN03Topp; pass2 is just
  49.         straightforward translation. 
  50.  
  51. 6/22/85:    Cleanup after initial debugging. Ready to
  52.         distribute. 
  53.  
  54. 10/8/85:    Adding \special's to imitate Textset's DVIAPS program.
  55.         Corrected bug in parsing of \specials. Version 1.
  56.  
  57. 10/15/85:   More changes for specials. Version 2.
  58.  
  59. 11/8/85:    Fixed bug in ln03:defpoint \special. Version 3.
  60.  
  61. 11/15/85:   Changed way information about font size is passed
  62.         back by add_txf_to_lnf. Add PXL-reading
  63.         capability. Version 4. 
  64.  
  65. 11/21/85:   Error handling has been reformed. Still version 4.
  66.  
  67. 12/12/85:   Bug correction in handling of plotfile special. 
  68.         Version 5.
  69.  
  70. 1/7/86:        Clear all fonts from memory at beginning of 
  71.         document. Version 6.
  72.  
  73. 1/16/86:    Fixed bug in dvi2ln3nft.c, nftf was not being
  74.         closed after use. Version 7. (Bug found by Mark
  75.         DeVries.) 
  76.  
  77. 3/4/86:        Support for landscape added -- very easy, if only
  78.         I had known. Fixed bug found by Mark DeVries,
  79.         error returned if \special selects device other
  80.         than LN03. Version 8. 
  81.  
  82. 3/28/86:    Conditionalization for Ultrix. #ifndef vms ==
  83.         #ifdef ultrix. Still emits 8-bit characters.
  84.         Should be unchanged under VMS. getenv is used to
  85.         replace logicals. Version 9. 
  86.  
  87. 4/4/86:     Added support for 7-bit only environments, 
  88.         conditionalized on the SEVENBIT symbol. Still 
  89.         version 9.
  90.  
  91. 4/16/86:    Final changes to make compilation under Ultrix 
  92.         clean. Still version 9.
  93.  
  94. 6/27/86:    Finally added option for A4-sized paper. Version 
  95.         10.
  96.  
  97. 11/6/86:    Bug when /n= is specified and /s= is not. Fixed
  98.         rule computation. Version 11.
  99.  
  100. 02/20/87:   Clean up messages.  Add PK file support to DVI2LN3NFT.C.
  101.         Genericize the PK file support.  Rename from Dvi2ln3 to
  102.         Ln03DVI to keep this version indepedent of the original.
  103.         Prepare/cleanup for realease on the Unix-Tex distribution.
  104.  
  105. 5/18/87:    Move version info to a seperate file, ln03version.c.
  106. */ 
  107.  
  108. #ifdef vms
  109. #include stdio
  110. #include ctype
  111. #else
  112. #include <stdio.h>
  113. #include <ctype.h>
  114. #endif
  115.  
  116. /* Ultrix seems to want this: */
  117.  
  118. #ifdef bsd4_2
  119. # define strchr index 
  120. extern int errno;
  121. #endif
  122.  
  123. #ifndef vms
  124. char *strchr(), *malloc(), *getenv();
  125. # define SEVENBIT 1
  126. #endif
  127.  
  128. /* The version information */
  129.  
  130. extern char ln03dvi_version[];
  131.  
  132.  
  133. /* In VMS, we declare all global variables to be globaldef.
  134. This is not really necessary, just an old habit from working
  135. with other VMS languages. It's done with #define's, so it
  136. may easily be undone when porting the program to other
  137. systems. */ 
  138.  
  139. #ifdef vms
  140. #define GLOBAL globaldef
  141. #define EXTERN globalref
  142. #else
  143. #define GLOBAL 
  144. #define EXTERN extern
  145. #endif
  146.  
  147.  
  148. /* Here begins a long list of global variables. */
  149.  
  150. /* hoff is the horizontal offset in pixels to be added to
  151. all dimensions read in; voff is the corresponding vertical
  152. offset. */ 
  153.  
  154. GLOBAL int hoff, voff;
  155. GLOBAL FILE *tfmfile,*outfile;
  156. GLOBAL int dvifile;
  157.  
  158. /* [[We are finally implementing the speed optimization of
  159. eliminating getc's on dvifile. Hence the following:]] */
  160.  
  161. GLOBAL unsigned char dvibuf[513];
  162. GLOBAL int dvifp,dvieof;
  163. #define mygetcdvi ((dvifp < 512) ? dvibuf[dvifp++] : \
  164.     moredvi()) 
  165.  
  166. /* Each page in a DVI file is identified by ten longwords,
  167. which appear immediately after each bop (beginning of page)
  168. command. Ln03DVI, like DVItype, supports certain options
  169. that allow one to print only selected pages. The following
  170. variables are used for this purpose. 
  171.  
  172. Num_pages is a count of how many pages have been passed to
  173. the output. Max_pages is the user-specified maximum number
  174. of pages to pass. How_many_counts denotes the number of
  175. identifying longwords to take into account when searching
  176. for the user-specified starting page. The user specifies the
  177. starting page as the first one whose identifying longwords
  178. match certain specified values, stored in start_page.
  179. However, identifying longword i is only required to match
  180. the value in start_page[i] if use_count[i] is nonzero. */ 
  181.  
  182. GLOBAL long int num_pages,max_pages;
  183. GLOBAL int how_many_counts,use_count[10];
  184. GLOBAL long int start_page[10];
  185.  
  186. /* When the output file can have no characters with code >
  187. 127, the user should employ a LN03DVI which is recompiled
  188. with the symbol SEVENBIT defined. Such a Ln03DVI will run slower
  189. than one which is allowed eight bits, and produce larger
  190. output files. */ 
  191.  
  192. #ifdef SEVENBIT
  193. GLOBAL int right7; 
  194. #endif
  195.  
  196. /* FILESPECLEN is the maximum size of a file specification
  197. under VMS. [[Perhaps arrays of fixed size should not be
  198. used, but rather malloc should be employed...]] */ 
  199.  
  200. #define FILESPECLEN 252
  201.  
  202. /* landscape says whether we are printing in landscape or
  203. portrait. */ 
  204.  
  205. GLOBAL char landscape;
  206.  
  207. /* a4_paper says whether we are printing on a4 paper. */
  208.  
  209. GLOBAL char a4_paper;
  210.  
  211. /* The main routine deals with the command line arguments,
  212. opens the dvi file, and then calls various other routines to
  213. handle the "passes". There are two passes, each of which
  214. involves reading the DVI file from start to finish; in
  215. between, the font load is constructed. */ 
  216.  
  217. main(argc,argv) 
  218. int argc;
  219. char *argv[];
  220. {
  221.     int status,i,jnam,jext;
  222.     char infnam[FILESPECLEN];
  223.  
  224.     if (argc < 2) {
  225.     printf("Usage:\tln03dvi [options] dvifile\n");
  226.     printf("    -h hoffset\t\t(horizontal offset in pixels)\n");
  227.     printf("    -v voffset\t\t(vertical offset in pixels)\n");
  228.     printf("    -s XYX\t\t(starting page number)\n");
  229.     printf("    -n pages\t\t(number of pages)\n");
  230.     printf("    -l\t\t\t(landscape mode)\n");
  231.     printf("    -e\t\t\t(european A4 paper)\n");
  232.     goto exit_label;
  233.     }
  234.  
  235.     printf("\n\tLn03DVI %s\n\n",ln03dvi_version);
  236.  
  237. /* Now decipher the options off the command line. */
  238.  
  239. #ifdef SEVENBIT
  240.     right7 = 0;
  241. #endif
  242.     max_pages = 1000000000;
  243.     how_many_counts = 0;
  244.     hoff = 300;
  245.     voff = 300;
  246.     landscape = 0;
  247.     a4_paper = 0;
  248.     for (i=1; i<argc; i++) {
  249.  
  250.     /* The extension of the input file defaults to .dvi. Locate
  251.     the filename field within the filespec parameter, set jname
  252.     to point to beginning, jext to point to one place after end.
  253.     [[This code is of course dependent on the VMS filespec
  254.     syntax.]] */ 
  255.  
  256.     if (argv[i][0] == '-') {
  257.         i += command_line_option( &argv[i], argc-i );
  258.     } else {
  259.         strcpy(infnam,argv[i]);
  260.  
  261.         /* The extension of the input file defaults to .dvi. Locate
  262.         the filename field within the filespec parameter, set jname
  263.         to point to beginning, jext to point to one place after end.
  264.         [[This code is of course dependent on the VMS filespec
  265.         syntax.]] */ 
  266.       
  267.         find_filename(infnam,&jnam,&jext);
  268.         if (infnam[jext] == '\0') strcat(infnam,".dvi");
  269.         dvifile = open(infnam,0);
  270.         if (dvifile < 0) {
  271.         fprintf(stderr,"Couldn't open dvi file ");
  272.         perror(infnam);
  273.         goto exit_label;
  274.         }
  275.         dvieof = 0;
  276.         dvifp = 512;
  277.     }
  278.     }
  279.  
  280. /* Now do the passes */
  281.  
  282.     status = pass1();
  283.     if (status != 0) goto exit_label; 
  284. #ifdef vms || bsd4_2
  285.     if (lseek(dvifile,0,0) == -1) {
  286.     printf("\n Couldn't rewind dvi file.");
  287.     goto exit_label;
  288.     }
  289. #else
  290.     lseek(dvifile,0,0);
  291. #endif
  292.     dvieof = 0;
  293.     dvifp = 512;
  294.  
  295.     status = open_output_file(infnam,jnam,jext,".ln3");
  296.     if (status != 0) { perror("Couldn't open output file");
  297.     goto exit_label; }
  298.  
  299.     status = make_font_load();
  300.     if (status != 0) goto exit_label;
  301.  
  302.     pass2();
  303.     printf("\n");
  304.  
  305.     fprintf(outfile,"\033[!p");
  306.     close(dvifile);
  307.     fclose(outfile);
  308.  
  309. exit_label:
  310. /*
  311. #ifndef vms
  312.     printf("\n");
  313. #endif
  314. */
  315.     ;
  316. }
  317.  
  318. /* Find_filename finds the filename part of a filespec
  319. passed in s, returning the index of the first character in
  320. *ns, and the index of the character after the last in *ne.
  321. [[This works under VMS and Unix. MS-DOS still needs work.]]
  322. */ 
  323.  
  324. int find_filename(s,ns,ne)
  325. char s[];
  326. int *ns,*ne;
  327. {
  328.     int jnam,jext,j,slen;
  329.  
  330.     slen = strlen(s);
  331.     jnam = 0;
  332.     for (j = slen-1; j >= 0; j--) {
  333. #ifdef vms
  334.     if (s[j] == ':' || s[j] == ']' ||
  335.         s[j] == '>') {
  336. #else
  337.     if (s[j] == '/') {
  338. #endif
  339.         jnam = j+1;
  340.         break;
  341.     }
  342.     }
  343.  
  344.     jext = slen;    
  345.     for (j = jnam; j < slen; j++) {
  346. #ifdef vms
  347.     if (s[j] == '.' || s[j] == ';') {
  348. #else
  349.     if (s[j] == '.') {
  350. #endif
  351.         jext = j;
  352.         break;
  353.     }
  354.     }
  355.  
  356.     *ns = jnam;
  357.     *ne = jext;
  358. }
  359.  
  360. /* Moredvi reads up to 512 bytes from the dvi file and puts
  361. them in the dvibuf. It also returns the first byte. Any
  362. missing bytes are filled in with 248, the dvi command code
  363. for the postamble. 
  364.  
  365. [[The read is not retried. Thus, the dvifile must be either
  366. stream or fixed-length 512 byte records for this to work.
  367. TeX-generated dvi files have fixed-length 512 byte
  368. records.]] */ 
  369.  
  370. unsigned char moredvi() {
  371.     int i,j;
  372.  
  373.     i = read(dvifile,dvibuf,512);
  374.     dvifp = 1;
  375.     if (i <= 0) {
  376.     dvieof = 1;
  377.     for (j=0; j<512; j++) dvibuf[j] = 248;
  378.     return(248);
  379.     } else if (i < 512) 
  380.     for (j=i; j<512; j++) dvibuf[j] = 248;
  381.     return(dvibuf[0]);
  382. }
  383.  
  384.  
  385. /* ERROR HANDLING: Errors should be reported as close to the
  386. source as possible, so that the maximum amount of
  387. information is available to the user to identify the error. 
  388.  
  389. Errors in the format of DVI or TFM files are not reported
  390. specifically, since there exist programs, DVItype and
  391. TFtoPL, which diagnose errors in such files. This program
  392. accepts some incorrect DVI files, for example, those with a
  393. bad postamble or bad backpointers. [[However, someday we may
  394. rewrite the program to select pages by using these features
  395. of the DVI format, rather than by skipping unwanted pages.]]
  396. */ 
  397.  
  398. GLOBAL char *bad_DVI_message = "Bad DVI file - check it with DVItype\n"; 
  399.  
  400. /* Command_line_option reads and processes options off the
  401. argument list. At this time, six options are supported. 
  402.  
  403. The N option sets the maximum number of pages to be printed.
  404. Its syntax is /N=<integer> . 
  405.  
  406. The H option modifies the default horizontal offset.
  407.  
  408. The V option modifies the default vertical offset. 
  409.  
  410. The S option has syntax /S=<pagespec>{.<pagespec>}* where
  411. the {}* denotes repetition, and a <pagespec> is either an
  412. integer or * to indicate any value. An example would be
  413. "/S=*.8.3". 
  414.  
  415. The meaning of the S option is as follows: In a DVI file,
  416. pages are identified by ten longword values which follow
  417. each bop (beginning of page) command. The value of the S
  418. option indicates which page to start printing on. For
  419. example, "*.8.3" means start printing at the first page
  420. whose second identifying longword is 8 and third is 3. 
  421.  
  422. The L option says print in landscape.
  423.  
  424. The E (for European) option says print on A4 paper. */ 
  425.  
  426. int command_line_option( argv, argc )
  427. char **argv;
  428. int argc;
  429. {
  430.     long int i = 0, k;
  431.     char *u;
  432.     char *t = &argv[0][1];
  433.  
  434.     if (toupper(t[0]) == 'N') {
  435.     if (isdigit(t[1]))
  436.         sscanf(&t[2], "%ld", &k);
  437.     else {
  438.         sscanf(argv[1], "%ld", &k );
  439.         i += 1;
  440.     }
  441.     max_pages = k;
  442.     } else if (toupper(t[0]) == 'H') {
  443.     if (isdigit(t[1]))
  444.         sscanf(&t[2], "%ld", &k);
  445.     else {
  446.         sscanf(argv[1], "%ld", &k );
  447.         i += 1;
  448.     }
  449.     hoff = k;
  450.     } else if (toupper(t[0]) == 'V') {
  451.     if (isdigit(t[1]))
  452.         sscanf(&t[2], "%ld", &k);
  453.     else {
  454.         sscanf(argv[1], "%ld", &k );
  455.         i += 1;
  456.     }
  457.     voff = k;
  458.     } else if (toupper(t[0]) == 'L') {
  459.     landscape = 1;
  460.     } else if (toupper(t[0]) == 'E') {
  461.     a4_paper = 1;
  462.     } else if (toupper(t[0]) == 'S') {
  463.     if (t[1] == 0) {
  464.         t = argv[1];
  465.         i++;
  466.     }
  467.     how_many_counts = 0;
  468.     for (;;) {
  469.         if (t[0] == '*') {
  470.         use_count[how_many_counts] = 0;
  471.         how_many_counts++;
  472.         } else if (sscanf(t,"%ld",&k) != 0) {
  473.         use_count[how_many_counts] = 1;
  474.         start_page[how_many_counts] = k;
  475.         how_many_counts++;
  476.         } else break;
  477.         u = strchr(t,'.');
  478.         if (u == 0) break;
  479.         t = &u[1];
  480.     }
  481.     }
  482.     return(i);
  483. }
  484.  
  485. /* Open_output_file opens one of the output files, using the
  486. file pointer outfile. The name of the output file is
  487. obtained by appending the string ext to the substring of
  488. infnam beginning at jnam and ending at jext. 
  489.  
  490. [[This code contains a VMS dependency. We use creat followed
  491. by fdopen to open the file as a normal VMS file
  492. ("rat=cr","rfm=var") rather than a STREAM_LF file. Beginning
  493. with VAX C V2.0, this can be done by just calling fopen.]]
  494. */ 
  495.  
  496. int open_output_file(infnam,jnam,jext,ext)
  497. char *infnam,*ext;
  498. int jnam,jext;
  499. {
  500.     char outfnam[FILESPECLEN];
  501.     int jj;
  502.  
  503.     strcpy(outfnam,&infnam[jnam]);
  504.     strcpy(&outfnam[jext-jnam],ext);
  505.  
  506. #ifdef vms
  507.     jj = creat(outfnam,0,"rat=cr","rfm=var");
  508. #else
  509.     jj = creat(outfnam,0644);
  510. #endif
  511.     if (jj == -1) return(1);
  512.     outfile = fdopen(jj,"w");
  513.     if (outfile == NULL) return(1);
  514.  
  515.     return(0);
  516. }
  517.  
  518. /* We use an overlay, as suggested in the DVItype
  519. documentation, to combine these bytes into larger integers.
  520. [[Note that this technique relies on the fact that numbers
  521. on the VAX are stored with the least significant byte first.
  522. The macros below would have to be changed if the program
  523. were to be ported to a machine architecture for which this
  524. is not so.]] 
  525.  
  526. [[The use of getc is expensive, since getc is a function in
  527. VAX C V2.0. Eventually, one would want to rewrite these
  528. macros to work like the old getc macro.]] */ 
  529.  
  530. #ifdef BIG_ENDIAN
  531. /* For Sun's */
  532. #define two_bytes_u lcx.uc[2] = mygetcdvi; \
  533.     lcx.uc[3] = mygetcdvi; lcx.uc[1] = 0; lcx.uc[0] = 0
  534.  
  535. #define two_bytes_s lcx.c[2] = mygetcdvi; \
  536.     lcx.c[3] = mygetcdvi;  \
  537.     if (lcx.c[2] >= 0) { lcx.uc[1] = 0; lcx.uc[0] = 0 ;} \
  538.     else { lcx.uc[1] = 255; lcx.uc[0] = 255; }
  539.  
  540. #define three_bytes_u lcx.uc[1] = mygetcdvi; lcx.uc[2] = mygetcdvi; \
  541.     lcx.uc[3] = mygetcdvi; lcx.uc[0] = 0
  542.  
  543. #define three_bytes_s lcx.c[1] = mygetcdvi; lcx.c[2] = mygetcdvi; \
  544.     lcx.c[3] = mygetcdvi;  \
  545.     lcx.uc[0] = (lcx.c[1] >= 0) ? 0 : 255;
  546.  
  547. #define four_bytes lcx.c[0] = mygetcdvi; \
  548.     lcx.c[1] = mygetcdvi; lcx.c[2] = mygetcdvi; \
  549.     lcx.c[3] = mygetcdvi; 
  550.  
  551. #else  /* Not big endian */
  552.  
  553. GLOBAL union lc { long int l; unsigned long int ul; char c[4]; 
  554.         unsigned char uc[4]; } lcx;
  555.  
  556. #define two_bytes_u lcx.uc[1] = mygetcdvi; \
  557.     lcx.uc[0] = mygetcdvi; lcx.uc[2] = 0; lcx.uc[3] = 0
  558. #define two_bytes_u lcx.uc[1] = mygetcdvi; \
  559.     lcx.uc[0] = mygetcdvi; lcx.uc[2] = 0; lcx.uc[3] = 0
  560.  
  561. #define two_bytes_s lcx.c[1] = mygetcdvi; \
  562.     lcx.c[0] = mygetcdvi;  \
  563.     if (lcx.c[1] >= 0) { lcx.uc[2] = 0; lcx.uc[3] = 0 ;} \
  564.     else { lcx.uc[2] = 255; lcx.uc[3] = 255; }
  565.  
  566. #define three_bytes_u lcx.uc[3] = mygetcdvi; lcx.uc[1] = mygetcdvi; \
  567.     lcx.uc[0] = mygetcdvi; lcx.uc[2] = 0
  568.  
  569. #define three_bytes_s lcx.c[2] = mygetcdvi; lcx.c[1] = mygetcdvi; \
  570.     lcx.c[0] = mygetcdvi; \
  571.     lcx.uc[3] = (lcx.c[2] >= 0) ? 0 : 255;
  572.  
  573. #define four_bytes lcx.c[3] = mygetcdvi; \
  574.     lcx.c[2] = mygetcdvi; lcx.c[1] = mygetcdvi; \
  575.     lcx.c[0] = mygetcdvi; 
  576.  
  577. #endif
  578.  
  579. /* Knuth's programs like to hardcode fixed maximum sizes for
  580. various things, for example, the maximum number of fonts
  581. allowed in a DVI file. In general, it is preferable to use
  582. the C function malloc to allocate storage as needed. That is
  583. what we generally do in this Ln03DVI, but there are some
  584. residues of the Knuthian approach, like MAXTEXFONTS below.
  585. By the way, we never attempt to return any storage to the
  586. system. 
  587.  
  588. [[Eventually, it would be better to get rid of MAXTEXFONTS
  589. and use a linked list of records instead. There would be no
  590. cpu time penalty to using a linked list, because the
  591. function set_curf below does a linear search through the
  592. font array anyway.]] 
  593.  
  594. Txf is a data structure describing a TeX font. */
  595.  
  596. #define MAXTEXFONTS 100
  597.  
  598. struct txf { unsigned char chu[256]; int bc, ec; long int space, design_size,
  599.     scaled_size; int nchs; };
  600.  
  601. GLOBAL struct txf *txfa[MAXTEXFONTS+1];
  602.  
  603. /* Font_name points to strings containing TeX font names.
  604. Those strings are created with malloc as needed. */ 
  605.  
  606. GLOBAL char *font_name[MAXTEXFONTS+1]; 
  607.  
  608. /* Font width information needs to be read from TFM files.
  609. TFM is a special format defined by TeX. "Ln03DVI" stores each
  610. width in a longword. DVItype tries to save space by a
  611. two-level width storage method. We just allocate an array of
  612. widths for each font with malloc. 
  613.  
  614. [[We don't check that font checksums match in Ln03DVI,
  615. because there are a lot of slightly obsolete TFMs floating
  616. around which would result in a checksum error, but seem to
  617. give perfectly reasonable formatted output nonetheless.]] */
  618.  
  619. GLOBAL long int *font_width[MAXTEXFONTS+1];
  620.  
  621. /* TeX fonts are referred to by their internal numbers,
  622. which go from 0 to nf-1. The DVI file refers to them by
  623. external numbers, hence the array to_ext used to convert
  624. internal numbers to external numbers. Curf is the internal
  625. font number of the current font in the DVI file. Curchu
  626. points to the 'chu' (character used) array of the current
  627. font. */ 
  628.  
  629. GLOBAL int to_ext[MAXTEXFONTS+1],
  630.     nf,
  631.     curf;
  632. GLOBAL unsigned char *curchu;
  633.  
  634. /* In some switch statements, a lot of cases have to be
  635. enumerated. We employ the following Knuthian macro for that
  636. purpose: */ 
  637.  
  638. #define four_cases(_x1) case _x1: case _x1+1: case _x1+2: case _x1+3: 
  639.  
  640. /* We define a lot of constants corresponding to the DVI
  641. operation codes. These are copied from DVItype. */ 
  642.  
  643. #define id_byte 2    /* current version of the dvi format */
  644. #define set_char_0 0 /* typeset character 0 and move right */
  645. #define set1 128     /* typeset a character and move right */
  646. #define set_rule 132 /* typeset a rule and move right */
  647. #define put1 133     /* typeset a character */
  648. #define put_rule 137 /* typeset a rule */
  649. #define nop 138      /* no operation */
  650. #define bop 139      /* beginning of page */
  651. #define eop 140      /* ending of page */
  652. #define push 141     /* save the current positions */
  653. #define pop 142      /* restore previous positions */
  654. #define right1 143   /* move right */
  655. #define w0 147       /* move right by |w| */
  656. #define w1 148       /* move right and set |w| */
  657. #define x0 152       /* move right by |x| */
  658. #define x1 153       /* move right and set |x| */
  659. #define down1 157    /* move down */
  660. #define y0 161       /* move down by |y| */
  661. #define y1 162       /* move down and set |y| */
  662. #define z0 166       /* move down by |z| */
  663. #define z1 167       /* move down and set |z| */
  664. #define fnt_num_0 171 /* set current font to 0 */
  665. #define fnt1 235     /* set current font */
  666. #define xxx1 239     /* extension to dvi primitives (\special) */
  667. #define xxx4 242     /* potentially long extension to dvi primitives */
  668. #define fnt_def1 243 /* define the meaning of a font number */
  669. #define pre 247      /* preamble */
  670. #define post 248     /* postamble beginning */
  671. #define post_post 249 /* postamble ending */
  672. #define undefined_command 250
  673.  
  674. GLOBAL long int mag, num, den;
  675. GLOBAL double conv, unmag_conv;
  676.  
  677. /* Read_preamble reads the preamble of the dvi file. The
  678. comment is thrown away. The format version number is
  679. checked. The magnification, numerator, and denominator are
  680. remembered in the globals mag, num, den. The float values
  681. conv and unmag_conv serve to convert measurements from DVI
  682. units to pixels. */ 
  683.  
  684. int read_preamble() {
  685.     unsigned int i;
  686.     int j;
  687.  
  688.     i = mygetcdvi;
  689.     if (i != pre) return(1);
  690.     i = mygetcdvi;
  691.     if (i != id_byte) return(1);
  692.     four_bytes; num = lcx.l;
  693.     if (num <= 0) return(1);
  694.     four_bytes; den = lcx.l;
  695.     if (den <= 0) return(1);
  696.     four_bytes; mag = lcx.l;
  697.     if (mag <= 0) return(1);
  698.  
  699.     unmag_conv = (num/254000.0) * (300.0/den);
  700.     conv = unmag_conv * (mag/1000.0);
  701.  
  702. /* Skip over the comment field. */
  703.     
  704.     i = mygetcdvi;
  705.     for (j=0; j<i; j++) mygetcdvi;
  706.     return(0);
  707. }
  708.  
  709. /* The dvi file format requires us to keep track of various
  710. quantities, among them the horizontal and vertical positions
  711. h and v, and four values referred to simply as x, y, z and
  712. w. We also keep pixel-rounded versions of h and v, called hh
  713. and vv. */ 
  714.  
  715. GLOBAL long h, v, x, y, z, w;
  716. GLOBAL int hh, vv;
  717.  
  718. /* The following function, copied from DVItype, reads from
  719. the DVI file the first parameter of the current DVI command.
  720. In the case of functions whose first parameter is implicit,
  721. such as set_char_0 through set_char_127, the implicit
  722. parameter is returned. */ 
  723.  
  724. long first_par(o)
  725. int o;
  726. {
  727.     unsigned int i;
  728.  
  729.     if ((o >= set_char_0) && (o < set_char_0+128))
  730.     return(o-set_char_0);
  731.     if ((o >= fnt_num_0) && (o < fnt_num_0+64))
  732.     return(o-fnt_num_0);
  733.     switch (o) {
  734.     case set1:
  735.     case put1:
  736.     case fnt1:
  737.     case xxx1:
  738.     case fnt_def1:
  739.         i = mygetcdvi;
  740.         return(i);
  741.     case set1+1:
  742.     case put1+1:
  743.     case fnt1+1:
  744.     case xxx1+1:
  745.     case fnt_def1+1:
  746.         two_bytes_u;
  747.         return(lcx.ul);
  748.     case set1+2:
  749.     case put1+2:
  750.     case fnt1+2:
  751.     case xxx1+2:
  752.     case fnt_def1+2:
  753.         three_bytes_u;
  754.         return(lcx.ul);
  755.     case right1:
  756.     case w1:
  757.     case x1:
  758.     case down1:
  759.     case y1:
  760.     case z1:
  761.         return(mygetcdvi);
  762.     case right1+1:
  763.     case w1+1:
  764.     case x1+1:
  765.     case down1+1:
  766.     case y1+1:
  767.     case z1+1:
  768.         two_bytes_s;
  769.         return(lcx.l);
  770.     case right1+2:
  771.     case w1+2:
  772.     case x1+2:
  773.     case down1+2:
  774.     case y1+2:
  775.     case z1+2:
  776.         three_bytes_s;
  777.         return(lcx.l);
  778.     case set_rule:
  779.     case put_rule:
  780.     case right1+3:
  781.     case w1+3:
  782.     case x1+3:
  783.     case down1+3:
  784.     case y1+3:
  785.     case z1+3:
  786.     case set1+3:
  787.     case put1+3:
  788.     case fnt1+3:
  789.     case xxx1+3:
  790.     case fnt_def1+3:
  791.         four_bytes;
  792.         return(lcx.l);
  793.     case w0:
  794.         return(w);
  795.     case x0:
  796.         return(x);
  797.     case y0:
  798.         return(y);
  799.     case z0:
  800.         return(z);
  801.     default:
  802.         return(0);
  803.     }
  804. }
  805.  
  806. /* Pass1 reads from the DVI file until the last page that
  807. has to be processed, as determined by the page selection
  808. data structures. Font definitions are processed, as are font
  809. selections; after the starting page begins, pass1 records
  810. which characters are used from which fonts. Other commands
  811. are ignored in pass1. Return 1 if something goes wrong, e.g.
  812. if the DVI file comes to a premature end. This resembles
  813. skip_pages below. */ 
  814.  
  815. int pass1() {
  816.  
  817.     unsigned int k;
  818.     long int p;
  819.     int i;
  820.     char startp;
  821.     
  822.     if (read_preamble() != 0) {
  823.     fprintf(stderr,bad_DVI_message);
  824.     return(1);
  825.     }
  826.  
  827.     num_pages = 0;
  828.  
  829.     for (;;) {
  830.     if (dvieof) {
  831.         fprintf(stderr,bad_DVI_message);
  832.         return(1);
  833.     }
  834.     k = mygetcdvi;
  835.     if (k == post) return(0);
  836.     if (k >= set_char_0 && k < set_char_0+128) {
  837.         p = k;
  838.         k = set1;
  839.         } else if (k >= fnt_num_0 && k < fnt_num_0+64) {
  840.         p = k-fnt_num_0;
  841.         k = fnt1;
  842.         } else p = first_par(k);
  843.  
  844.     switch (k) {
  845.         case bop:
  846.         startp = 1;
  847.         for (i=0; i<10; i++) {
  848.             four_bytes;
  849.             if (i < how_many_counts && use_count[i] &&
  850.                 lcx.l != start_page[i]) startp = 0;
  851.             }
  852.         four_bytes;
  853.         if (startp != 0 || num_pages > 0) {
  854.             num_pages++;
  855.             if (num_pages > max_pages) return(0);
  856.         }
  857.         break;
  858.         case set_rule:
  859.         case put_rule:
  860.         four_bytes;
  861.             break;
  862.         four_cases(fnt_def1)
  863.         i = define_font(p);
  864.         if (i != 0) return(1);
  865.         break;
  866.         four_cases(fnt1)
  867.         if (num_pages > 0) set_curf(p);
  868.         break;
  869.             four_cases(set1)
  870.         four_cases(put1)
  871.         if (num_pages > 0)
  872.             txfa[curf] -> chu[p] = 1;
  873.         break;
  874.  
  875. /* \special's are currently ignored in pass1, and handled
  876. only in pass2. [[This will change if the putchar special is
  877. implemented.]] */ 
  878.  
  879.         four_cases(xxx1)
  880.         for (; p>0; p--) mygetcdvi;
  881.         break;
  882.             default:
  883.         break;
  884.     }
  885.     }
  886. /*  return(0); /* Unreachable */
  887. }
  888.  
  889. /* FONT LOAD BUILDING: The hardest thing Ln03DVI has to do
  890. is to build an LN03 font load that contains the glyphs
  891. required to print the DVI file. The following code pertains
  892. to that effort. 
  893.  
  894. LN03 fonts come in two flavors. Left fonts are invoked by
  895. character codes 33 to 126. Right fonts are invoked by
  896. character codes 161 to 254. Hence: */ 
  897.  
  898. #define leftfirst 33
  899. #define rightfirst 161
  900. #define leftlast 126
  901. #define rightlast 254
  902.  
  903. /* It is believed that there is a maximum of 32
  904. downline-loaded LN03 fonts. We always deal with these in
  905. pairs, using one as a left font, the other as a right font.
  906. Hence we have a maximum of 16 pairs. */ 
  907.  
  908. #define maxnfonts 16
  909.  
  910. /* We keep track of the true last character in each LN03
  911. font pair; the width of each character; and the LN03 name
  912. assigned to that pair. [[Yikes. Chw is an unsigned char
  913. here; it should be at least an unsigned short.]] */ 
  914.  
  915. GLOBAL int lastch[maxnfonts];
  916. GLOBAL unsigned char chw[maxnfonts][256];
  917. GLOBAL char fname[maxnfonts][32];
  918.  
  919. /* Txf is a record structure describing a TeX font. Txfa is
  920. an array of txf records. Txf2lnf describes the mapping
  921. between TeX fonts and pairs of LN03 fonts. Because the
  922. constructed LN03 font load has just the glyphs that are
  923. needed, we can often cram more than one TeX font into a pair
  924. of LN03 fonts. */ 
  925.  
  926. GLOBAL int txf2lnf[MAXTEXFONTS];
  927.  
  928. /* Maxfontnos reflects the fact that LN03 fonts must be
  929. denoted by a number from 10 to 19 in order to be selected as
  930. the LN03's current font. */ 
  931.  
  932. #define maxfontnos 9
  933.  
  934. /* The LN03 font-denoting numbers 10-19 have to be allocated
  935. among the fonts. The useno array keeps track of which number
  936. a font is using, -1 if the font currently doesn't have any
  937. number. The whouses array says which font is using a given
  938. number. */ 
  939.  
  940. GLOBAL int useno[maxnfonts],whouses[maxfontnos];
  941.  
  942. /* After the font load we declare the right and bottom
  943. margins we are going to use; these are always the maximum
  944. allowed values. */ 
  945.  
  946. GLOBAL int maxrmar,maxbmar;
  947.  
  948. /* The global ras_len_added is employed by the
  949. font-load-creating code to communicate how many bytes of
  950. raster information it adds to the font load each time it is
  951. called. [[Maybe there should be no EXTERNs in the main code
  952. file, just GLOBALs?]] The right thing would be to put all
  953. these declarations in .h's.]] */ 
  954.  
  955. EXTERN long ras_len_added;
  956.  
  957. #define max(x,y) (((x)>(y))?(x):(y))
  958. #define min(x,y) (((x)<(y))?(x):(y))
  959.  
  960. /* Make_font_load calls on other procedures to generate the
  961. LN03 font load. It writes the opening lines of the output
  962. file, too. */ 
  963.  
  964. int make_font_load() {
  965.     
  966.     int i,j,jj,k,l,fno,maxfno,chsize;
  967.     long int totsize;
  968.     int lnfcnt,txfcnt,the_txf,txf_size,lnfleft;
  969.     int txford[MAXTEXFONTS];
  970.     char cnt[3];
  971.  
  972.     totsize = 0;
  973.     chsize = 0;
  974.     curf = 0;
  975.     for (i=0; i<maxnfonts; i++) 
  976.     for (j=0; j<256; j++) chw[i][j] = 0;
  977.  
  978.     for (jj=0; jj<MAXTEXFONTS; jj++) txf2lnf[jj] = -1;
  979.  
  980. /* Fill the fname strings with different valid LN03 font
  981. names. [[This initialization could be done statically, but
  982. then maxnfonts would be more hardcoded.]] */ 
  983.  
  984.     for (i=0; i<maxnfonts; i++) 
  985.         strcpy(fname[i],"U000000002SK00GG0001UZZZZ02F000");
  986.     for (jj=1; jj<maxnfonts; jj++) {
  987.     sprintf(cnt,"%02d",jj);
  988.     strncpy(&(fname[jj][5]),cnt,2);
  989.     }
  990.  
  991. /* The margins are set to values that seem appropriate for
  992. American 8 1/2 by 11" paper. It is not clear if this needs
  993. to be changed for European A4 paper. Not changing it might
  994. deprive Europeans of access to the bottom 1.5cm of their
  995. paper. 
  996.  
  997. I have tried to set the margins appropriately according
  998. to the paper type (a4 or 8 1/2 x 11"), and to throw away
  999. text which is set outside. [[However, text too close to the
  1000. bottom margin still results in an apparently undocumented
  1001. LN03 page eject.]] */ 
  1002.  
  1003.     if (landscape == 1) {
  1004.     if (a4_paper == 0) {
  1005.         maxrmar = 3300;
  1006.         maxbmar = 2475;    
  1007.     } else {
  1008.         maxrmar = 3475;
  1009.         maxbmar = 2475;    
  1010.     }
  1011.     } else {
  1012.     if (a4_paper == 0) {
  1013.         maxrmar = 2550;
  1014.         maxbmar = 3300;
  1015.     } else {
  1016.         maxrmar = 2550;
  1017.         maxbmar = 3475;    
  1018.     }
  1019.     } 
  1020.  
  1021. /* In the following, esc[?21 J means print in landscape on 8
  1022. 1/2 x 11 inch paper, esc[?22 J means print in portrait on A4
  1023. paper, esc[?23 J means landscape on A4 paper, esc[?27h means
  1024. "advance the carriage by the character width when you set a
  1025. character", esc[11h and esc[7 I together mean to interpret
  1026. all dimensions in escape sequences as pixel units; esc[?52h
  1027. means our origin of coordinates is the upper left edge of
  1028. the paper; esc[%dt means the "maximum length" of the paper
  1029. is maxbmar pixels. */ 
  1030.  
  1031.     fprintf(outfile,"\033[!p");
  1032.     if (landscape == 1 && a4_paper == 0) fprintf(outfile,"\033[?21 J");
  1033.     if (landscape == 0 && a4_paper == 1) fprintf(outfile,"\033[?22 J");
  1034.     if (landscape == 1 && a4_paper == 1) fprintf(outfile,"\033[?23 J");
  1035.     fprintf(outfile,"\033[?7l\033[?27h\033[11h\033[7 I\033[?52h\033[%dt\n",
  1036.     maxbmar);
  1037.  
  1038.     for (i=0; i<maxnfonts; i++) useno[i] = -1;
  1039.     for (i=0; i<maxfontnos; i++) whouses[i] = -1;
  1040.     maxfno = -1;
  1041.  
  1042. /* Write font loading escape sequence onto outf. The ;0y
  1043. causes all previously downloaded fonts to be cleared from
  1044. the LN03's font memory. */ 
  1045.  
  1046.     fprintf(outfile,"\033P0;1;0y");
  1047.  
  1048. /* Count how many glyphs each font requires. */
  1049.  
  1050.     for (fno=0; fno<nf; fno++) {
  1051.  
  1052.     txfa[fno] -> nchs = 0;
  1053.     for(i=0; i<256; i++) 
  1054.         if (txfa[fno] -> chu[i] != 0) txfa[fno] -> nchs++;
  1055.  
  1056.     if (txfa[fno] -> nchs > 188)  goto need_empty_slots;
  1057.     chsize += txfa[fno] -> nchs;
  1058.     }
  1059.     printf("Font load to contain %d glyphs.\n",chsize);
  1060.  
  1061.  
  1062. /* Now we have to allocate TeX fonts to LN03 font pairs and
  1063. perform the actual load. The goal is to use as few LN03
  1064. fonts as possible. 
  1065.  
  1066. I don't know any way to do the allocation optimally, so a
  1067. first-fit heuristic is used. 
  1068.  
  1069. 1.    Find the largest unassigned TeX font. If none, exit,
  1070.     we're done. Number of glyphs used is the measure of size. 
  1071.  
  1072. 2.    Allocate an LN03 font pair for that TeX font and put
  1073.     the TeX font into it. 
  1074.  
  1075. 3.    Find the largest remaining TeX font that fits in
  1076.     what is left of that LN03 font pair. 
  1077.  
  1078.     3.1. if none exists, go to 1. 
  1079.     3.2. if one exists, put it in the font pair and go back 
  1080.      to 3. 
  1081.  
  1082. The txford array contains the TeX font numbers in the order
  1083. that they are assigned. Txfcnt keeps track of how many TeX
  1084. fonts have been assigned so far. */ 
  1085.  
  1086.     lnfcnt = 0;
  1087.     txfcnt = 0;
  1088.     for (i=0; i<MAXTEXFONTS; i++) txford[i] = -1;
  1089.  
  1090.     while (1) {
  1091.  
  1092. /* Find largest unassigned TeX font */
  1093.  
  1094.     lnfleft = rightlast-rightfirst+1+leftlast-leftfirst+1;
  1095.     txf_size = -1;
  1096.     for (j = 0; j < MAXTEXFONTS; j++) {
  1097.         if (txfa[j] != 0 && txf2lnf[j] == -1 
  1098.         && txfa[j] -> nchs > txf_size) {
  1099.         txf_size = txfa[j] -> nchs;
  1100.         the_txf = j;
  1101.         }
  1102.     }
  1103.     if (txf_size <= 0)  goto assignment_done;
  1104.     if (lnfcnt > maxnfonts)  goto too_complex;
  1105.     if (txf_size > lnfleft)  goto need_empty_slots;
  1106.  
  1107. /* Now allocate new LN03 font pair and put the current TeX
  1108. font into it */ 
  1109.  
  1110.     txf2lnf[the_txf] = lnfcnt;
  1111.     txford[txfcnt] = the_txf;
  1112.     txfcnt++;
  1113.     k = leftfirst-1;
  1114.     for (j = 0; j <= 255; j++) {
  1115.         if (txfa[the_txf] -> chu[j] != 0) {
  1116.         k++;
  1117.         if (k == leftlast+1)  k = rightfirst;
  1118.         lnfleft--;
  1119.         txfa[the_txf] -> chu[j] = k; 
  1120.         }
  1121.     }
  1122.     lastch[lnfcnt] = k;
  1123.  
  1124. /* Now try to fill the remaining part of the LN03 font pair
  1125. using other TeX fonts */ 
  1126.  
  1127.     while (1) {
  1128.  
  1129.         txf_size = -1;
  1130.         for (j = 0; j < MAXTEXFONTS; j++) {
  1131.         if (txfa[j] != 0 && txf2lnf[j] == -1
  1132.             && txfa[j] -> nchs > txf_size 
  1133.             && txfa[j] -> nchs <= lnfleft) {
  1134.             txf_size = txfa[j] -> nchs;
  1135.             the_txf = j;
  1136.         }
  1137.         }
  1138.         if (txf_size <= 0)  break;
  1139.  
  1140.         txf2lnf[the_txf] = lnfcnt;
  1141.         txford[txfcnt] = the_txf;
  1142.         txfcnt++;
  1143.         k = lastch[lnfcnt];
  1144.         for (j=0; j<=255; j++) {
  1145.         if (txfa[the_txf] -> chu[j] != 0) {
  1146.             k++;
  1147.             lnfleft--;
  1148.             if (k == leftlast+1) k = rightfirst;
  1149.             txfa[the_txf] -> chu[j] = k; 
  1150.         }
  1151.         }
  1152.         lastch[lnfcnt] = k;
  1153.     }
  1154.     
  1155.     lnfcnt++;
  1156.     }
  1157.  
  1158. assignment_done:
  1159.  
  1160. /* At this point, the TeX fonts have been assigned to LN03
  1161. font pairs. The assignment is reflected in the arrays
  1162. txf2lnf, lastch and txford. It remains to actually generate
  1163. the desired font load. This has to be done carefully, since
  1164. the function add_txf_to_lnf only supports adding glyphs to
  1165. an LN03 font pair in ascending order of character code. */ 
  1166.  
  1167.     for (j=0; j<txfcnt; j++) {
  1168.     k = txford[j];
  1169.     if (j != 0 && txf2lnf[k] != txf2lnf[txford[j-1]])  
  1170.         fprintf(outfile,",\n");
  1171.     add_txf_to_lnf(txf2lnf[k],k);    
  1172.     totsize += ras_len_added;
  1173.     }
  1174.  
  1175. /* [[At this point we ought to add code that writes a
  1176. message saying how much font RAM the load will occupy. But
  1177. there is no documentation for determining that! So, we make
  1178. the following compromise:]] */ 
  1179.  
  1180.     printf("Approximate size of font load: %ld bytes\n",totsize);
  1181.  
  1182. /* Now output escape sequences to the LN03 that assign the
  1183. available font numbers to the first ten font pairs in the
  1184. load. [[We may want to insert a few \n's in the following to
  1185. keep the line from getting too long.]] */ 
  1186.  
  1187.     fprintf(outfile,"\n;Ln03DVI 12 font load\033\\");
  1188.     fprintf(outfile,"\033[1;%ds\033[%dt\033[1;%dr",maxrmar,maxbmar,maxbmar);
  1189.     for (j=0; j <= min(maxfontnos,lnfcnt-1); j++) {
  1190.     useno[j] = j;
  1191.     whouses[j] = j;
  1192.     k = fname[j][16];
  1193.     fname[j][16] = '\0';
  1194.     fprintf(outfile,"\033P1;1%d}%s\033\\",j,fname[j]);
  1195.     fname[j][16] = k;
  1196.     }
  1197.     fprintf(outfile,"\033[10m");
  1198.  
  1199.     return(0);
  1200.  
  1201. too_complex:
  1202.     fprintf(stderr,"Can't construct a font load:\n");
  1203.     fprintf(stderr,"Dvi file uses too many glyphs from too many different fonts.\n");
  1204.     return(1);
  1205.  
  1206. /* [[We need to determine what the real limit is on the
  1207. number of LN03 fonts, and refine the test to detect if it is
  1208. being exceeded. Fortunately, only a very complex DVI file
  1209. could exceed the limit.]] */ 
  1210.  
  1211. need_empty_slots:
  1212.     {
  1213.     int msg = (100.0*conv*txfa[fno] -> scaled_size)/
  1214.         (unmag_conv*txfa[fno] -> design_size) + 0.5;
  1215.  
  1216.     fprintf(stderr,"Font %s",font_name[fno]);
  1217.     if (msg != 100) printf(" (@ %d%%)",mag);
  1218.     fprintf(stderr," uses > 188 characters. Can't handle that.\n");
  1219.     return(1);
  1220.     }
  1221.  
  1222. /* [[The error message above should be fixed to specify the
  1223. magnification of the font also. -- done, matt]] */ 
  1224.  
  1225. }
  1226.  
  1227. /* The lines in the LN3 file are limited to lengths of about
  1228. 100 bytes. We keep track of how many bytes are written so
  1229. far in the global lnhp. The accounting is conservative, so
  1230. e.g. each integer representing a pixel position counts as 4
  1231. bytes even though it might be shorter. [[One could easily
  1232. increase 100 to 200. It is not clear what problems are
  1233. provoked by going over, say, 256.]] 
  1234.  
  1235. Vpset keeps track of whether the vertical position needs to
  1236. be output to the LN03 file before setting any more
  1237. characters. Hh_old keeps track of the horizontal position
  1238. which LN03 thinks it's at. */ 
  1239.  
  1240. GLOBAL int ln3p,vpset,hh_old;
  1241.  
  1242. #define inc_ln3p(x) ln3p += x; if (ln3p > 100) \
  1243.     { ln3p = 0; vpset = 0; hh_old = 30000; }
  1244.  
  1245. /* [[In the above macro, hh_old is set to 30000 to mean "a
  1246. very large number" (the maximum valid hh for an LN03 is
  1247. 2550), so as to force the code below to re-output the true
  1248. hh. What are the implications of this kludge? Would it not
  1249. be more reasonable to have instead an hhset variable?]] */ 
  1250.  
  1251. /* A stack of hvxyzw values is kept; the stack pointer is
  1252. called s. [[Again, it would be better to make this stack a
  1253. linked list.]] */ 
  1254.  
  1255. #define STACKSIZE 100
  1256.  
  1257. GLOBAL long hstack[STACKSIZE], vstack[STACKSIZE], xstack[STACKSIZE], 
  1258.     ystack[STACKSIZE], zstack[STACKSIZE], wstack[STACKSIZE];
  1259. GLOBAL int hhstack[STACKSIZE], vvstack[STACKSIZE];
  1260. GLOBAL int s;
  1261.  
  1262. /* DVI files specify distances in units of 2^-16 points.
  1263. When translating to a device-specific format, it is
  1264. necessary to round the DVI distances to pixel units. This is
  1265. done by means of the pixel_round macro. 
  1266.  
  1267. However, rather than using this macro in the straightforward
  1268. way, rounding is often performed by more elaborate
  1269. techniques, which we call "Stanford rules" (after John Le
  1270. Carre's "Moscow rules"). These rules make use of a parameter
  1271. MAX_DRIFT, which is roughly the maximum number of pixels
  1272. that things are allowed to deviate from straightforward
  1273. rounding. */ 
  1274.  
  1275. #define pixel_round(x) ((int) ((x<0) ? \
  1276.     (conv*(x)-0.5) : (conv*(x)+0.5)))
  1277.     
  1278. #define MAX_DRIFT 2
  1279.  
  1280. /* Pass2 reads the dvi file and interprets the commands in
  1281. it, usually by calling other routines. The interpretation
  1282. generally consists of writing something into the output
  1283. file, updating the current fonts and positions, and possibly
  1284. updating the stack. */ 
  1285.  
  1286. int pass2()
  1287. {
  1288.     int i,j;
  1289.     long int p;
  1290.     unsigned int k;
  1291.     
  1292.     if (read_preamble() != 0) {
  1293.     fprintf(stderr,bad_DVI_message);
  1294.     return(1);
  1295.     }
  1296.  
  1297.     curf = 0;
  1298.     ln3p = 0;
  1299.     s = 0;
  1300.     vpset = 0;
  1301.     hh_old = 30000;
  1302.  
  1303. /* Skip pages until the desired starting page is reached. A
  1304. nonzero value is returned if the starting page is never
  1305. encountered. */ 
  1306.  
  1307.     if (skip_pages() != 0) return(0); 
  1308.  
  1309.     for (;;) {
  1310.     if (dvieof) { 
  1311.             fprintf(stderr,bad_DVI_message);
  1312.         return(1); }
  1313.     k = mygetcdvi;
  1314.     if (k == post) return(0);
  1315.     if (k >= undefined_command) {
  1316.             fprintf(stderr,bad_DVI_message);
  1317.         return(1); }
  1318.     
  1319.     if (k >= set_char_0 && k < set_char_0+128) {
  1320.         p = k;
  1321.         k = set1;
  1322.         } else if (k >= fnt_num_0 && k < fnt_num_0+64) {
  1323.         p = k-fnt_num_0;
  1324.         k = fnt1;
  1325.         } else p = first_par(k);
  1326.  
  1327.     j = do_command(k,p);
  1328.     if (j == 2) return(0); /* done with the required number of pages */
  1329.     else if (j == 1) return(1); /* error encountered, stop */
  1330.  
  1331.     }
  1332. }
  1333.  
  1334. GLOBAL long first_counter;
  1335.  
  1336. /* Read from the DVI file until the starting page condition
  1337. is met. Return 1 if something goes wrong, or if the DVI file
  1338. comes to an end. This function is copied quite closely from
  1339. DVItype. */ 
  1340.  
  1341. int skip_pages() {
  1342.  
  1343.     unsigned int k;
  1344.     int i;
  1345.     long p;
  1346.     char startp;
  1347.     
  1348.     for (;;) {
  1349.     if (dvieof) return(1);
  1350.     k = mygetcdvi;
  1351.     if (k == post) return(1);
  1352.     if (k >= set_char_0 && k < set_char_0+128) {
  1353.         p = k;
  1354.         k = set1;
  1355.         } else if (k >= fnt_num_0 && k < fnt_num_0+64) {
  1356.         p = k-fnt_num_0;
  1357.         k = fnt1;
  1358.         } else p = first_par(k);
  1359.  
  1360.     switch (k) {
  1361.         case bop:
  1362.         startp = 1;
  1363.         for (i=0; i<10; i++) {
  1364.             four_bytes;
  1365.             if (i == 0) first_counter = lcx.l;
  1366.             if (i < how_many_counts && use_count[i] &&
  1367.                 lcx.l != start_page[i]) startp = 0;
  1368.             }
  1369.         four_bytes;
  1370.         if (startp != 0) { 
  1371.             v = 0; vv = 0; h = 0; hh = 0;
  1372.             printf("\n [%ld]",first_counter);
  1373.             num_pages = 1; return(0); }
  1374.         break;
  1375.         case set_rule:
  1376.         case put_rule:
  1377.         four_bytes;
  1378.             break;
  1379.         four_cases(fnt_def1)
  1380.         define_font_pass2(p);
  1381.         break;
  1382.         four_cases(xxx1)
  1383.         for (; p>0; p--) mygetcdvi;
  1384.         break;
  1385.             default:
  1386.         break;
  1387.     }
  1388.     }
  1389. /*  return(0); /* Unreachable */
  1390. }
  1391.  
  1392. /* Do_command performs the DVI command of code k, assuming
  1393. the "first parameter" is p. It is assumed that k is not one
  1394. of set_char0 through set_char127. */ 
  1395.  
  1396. int do_command(k,p)
  1397. int k;
  1398. long p;
  1399. {
  1400.     int i,j,l,lnf;
  1401.  
  1402.     switch (k) {
  1403.     four_cases(fnt_def1)
  1404.         define_font_pass2();
  1405.         break;
  1406.  
  1407. /* It is possible that the dvi file sets some glyphs off the
  1408. page. No error message is given, since the user will
  1409. generally see what is happening in the output (plus, if
  1410. Ln03DVI gives an error message, users will complain about
  1411. Ln03DVI even though the error is in their TeX file). 
  1412.  
  1413. In most cases, Ln03DVI takes no special action at all, since
  1414. the LN03 will clip the glyphs for us. Unfortunately, the
  1415. LN03 doesn't let us specify negative horizontal or vertical
  1416. positions, so we have to clip glyphs at such positions away
  1417. ourselves. Also, if one sets at a position below the bottom
  1418. margin, the LN03 will eject a page, so glyphs that are set
  1419. below that margin also have to be clipped by hand. 
  1420.  
  1421. The following code accomplishes that: */ 
  1422.  
  1423.     four_cases(put1)
  1424.     four_cases(set1)
  1425.  
  1426.         if (vv+voff > 0 && vv+voff <= maxbmar && hh+hoff > 0) {
  1427.         if (!vpset) {
  1428.             fprintf(outfile,"\n\033[%dd",vv+voff);
  1429.             fprintf(outfile,"\033[%d`",hh+hoff);
  1430.             ln3p = 16;
  1431.             vpset = 1;
  1432.             hh_old = hh;
  1433.         }
  1434.         if (hh_old != hh) {
  1435.             if (hh > hh_old) fprintf(outfile,"\033[%da",hh-hh_old);
  1436.             else fprintf(outfile,"\033[%d`",hh+hoff);
  1437.             ln3p += 7;
  1438.         }
  1439.  
  1440. #ifndef SEVENBIT
  1441.         putc(curchu[p],outfile);
  1442. #else 
  1443.         if (curchu[p] > 127) {
  1444.             if (right7 == 0) {
  1445.             fputs("\033n",outfile);
  1446.             inc_ln3p(2);
  1447.             right7 = 1;
  1448.             }
  1449.             putc(curchu[p]-128,outfile);
  1450.         } else {
  1451.             if (right7 == 1) {
  1452.                 putc(15,outfile);
  1453.             inc_ln3p(1);
  1454.             right7 = 0;
  1455.             }
  1456.             putc(curchu[p],outfile);
  1457.         }
  1458. #endif
  1459.         inc_ln3p(1);
  1460.         }
  1461.         if (k >= put1) {
  1462.             hh_old = hh+chw[txf2lnf[curf]][curchu[p]];
  1463.         break;
  1464.         }
  1465.         h += font_width[curf][p];
  1466.  
  1467. /* In rounding h to generate the pixel-position hh, Stanford
  1468. rules (see above) come into play. We set the new hh
  1469. (horizontal position in pixels) to the value obtained by
  1470. adding the pixel width of the character being set to the
  1471. current position. We then correct this value so that it does
  1472. not exceed the rounded version of the true position by more
  1473. than MAX_DRIFT pixels. 
  1474.  
  1475. Note that if we did not apply Stanford rules here, or
  1476. equivalently if we set MAX_DRIFT to zero, many more
  1477. set-X-position commands would appear in the output. */ 
  1478.  
  1479.         hh += chw[txf2lnf[curf]][curchu[p]];
  1480.         hh_old = hh;
  1481.         l = pixel_round(h);
  1482.         if (hh-l > MAX_DRIFT) hh = l+MAX_DRIFT;
  1483.         else if (l-hh > MAX_DRIFT) hh = l-MAX_DRIFT;
  1484.         break;
  1485.  
  1486.     four_cases(fnt1)
  1487.         set_curf(p);
  1488.         curchu = &(txfa[curf] -> chu[0]);
  1489.         lnf = txf2lnf[curf];
  1490.         if (useno[lnf] == -1) {
  1491.         useno[whouses[maxfontnos]] = -1;
  1492.         useno[lnf] = maxfontnos;
  1493.         whouses[maxfontnos] = lnf;
  1494.         fprintf(outfile,"\033P1;1%d}%16s\033\\",maxfontnos,
  1495.             fname[lnf]);
  1496.         inc_ln3p(26);
  1497.         }
  1498.         fprintf(outfile,"\033[1%dm",useno[lnf]);
  1499.         inc_ln3p(5);
  1500.         break;
  1501.  
  1502.     case set_rule:
  1503.     case put_rule: 
  1504.  
  1505. /* When converting rule dimensions to pixel dimensions, we
  1506. do not follow Stanford rules. Rather, we just round the true
  1507. positions to obtain the pixel positions. This avoids
  1508. unsightly gaps between rules. [[It should not cause much of
  1509. a problem with typical rule applications (ruled tables,
  1510. fraction bars)...but perhaps with large delimiters there
  1511. might be some difficulty. This needs more thought...]] */ 
  1512.  
  1513.         four_bytes; 
  1514.         if (p >= 0 && lcx.l >= 0) {
  1515.             i = pixel_round(p);
  1516.         j = pixel_round(lcx.l);
  1517.                 if (p > 0 && i == 0) i = 1;
  1518.                 if (lcx.l > 0 && j == 0) j = 1;
  1519.         do_rule(pixel_round(h),
  1520.             pixel_round(v)-i,pixel_round(h)+j,
  1521.             pixel_round(v));
  1522.             }
  1523.         if (k == set_rule) {
  1524.         h += lcx.l;
  1525.         hh = pixel_round(h);
  1526.         }
  1527.         break;
  1528.  
  1529.      case push:
  1530.         s++;
  1531.         if (s == STACKSIZE) {
  1532.         fprintf(stderr,"\nStack too deep for Ln03DVI\n");
  1533.         return(1);
  1534.         }
  1535.         xstack[s] = x;
  1536.         ystack[s] = y;
  1537.         vstack[s] = v;
  1538.         hstack[s] = h;
  1539.         vvstack[s] = vv;
  1540.         hhstack[s] = hh;
  1541.         wstack[s] = w;
  1542.         zstack[s] = z;
  1543.         break;
  1544.  
  1545.     case pop:
  1546.         if (s == 0) {
  1547.         fprintf(stderr,bad_DVI_message); 
  1548.         return(1); 
  1549.         }
  1550.         if (vv != vvstack[s]) vpset = 0;
  1551.         x = xstack[s]; 
  1552.         y = ystack[s];    
  1553.         v = vstack[s]; 
  1554.         h = hstack[s];    
  1555.         vv = vvstack[s]; 
  1556.         hh = hhstack[s];    
  1557.         w = wstack[s]; 
  1558.         z = zstack[s];
  1559.         s--; 
  1560.         break; 
  1561.  
  1562.     four_cases(xxx1)
  1563.         do_special_pass2(p);
  1564.         break;
  1565.  
  1566.     case bop:
  1567.  
  1568. /* If we've done the required number of pages, we'll skip
  1569. the rest of the DVI file. If not, type the first parameter
  1570. of bop on the user's terminal the way TeX does, to give an
  1571. indication of progress. */ 
  1572.  
  1573.         if (num_pages == max_pages) return(2);
  1574.         v = 0; vv = 0; h = 0; hh = 0;
  1575.         vpset = 0; hh_old = 100000;
  1576.         four_bytes;
  1577.         if (num_pages%12 == 0) printf("\n");
  1578.         first_counter = lcx.l;
  1579.         printf(" [%ld]",lcx.l);
  1580.         fflush(stdout);
  1581.         num_pages++;
  1582.         for (i = 0; i<40; i++) mygetcdvi; 
  1583.         break;
  1584.  
  1585.     case eop:
  1586.         fprintf(outfile,"\n\f");
  1587.         ln3p = 0;
  1588.         break;
  1589.  
  1590. /* Now we have to consider the cases for pure motion. */ 
  1591.  
  1592.     four_cases(right1)
  1593.         set_h(h+p);
  1594.         break;
  1595.     four_cases(x1)
  1596.         x = p;
  1597.     case x0:
  1598.         set_h(h+x);
  1599.         break;
  1600.     four_cases(y1)
  1601.         y = p;
  1602.     case y0:
  1603.         set_v(v+y);
  1604.         break;
  1605.     four_cases(w1)
  1606.         w = p;
  1607.     case w0:
  1608.         set_h(h+w);
  1609.         break;
  1610.     four_cases(z1)
  1611.         z = p;
  1612.     case z0:
  1613.         set_v(v+z);
  1614.         break;
  1615.     four_cases(down1)
  1616.         set_v(v+p);
  1617.         break;
  1618.  
  1619.     }    
  1620.     return(0);
  1621. }
  1622.  
  1623. /* Set_curf sets the current font to external number p. Note
  1624. that the current font is maintained as an internal font
  1625. number. */ 
  1626.  
  1627. int set_curf(p)
  1628. int p;
  1629. {
  1630.     to_ext[nf] = p;
  1631.     curf = 0;
  1632.     while (to_ext[curf] != p) { curf++; }    
  1633. }
  1634.  
  1635. /* Define_font processes a font definition from the DVI
  1636. file. The TFM file for the font is read at this point from
  1637. the directory TEX$FONTS.
  1638.  
  1639. [[Originally it made sense to read the DVI files here,
  1640. because we were supporting the LN03 font file format only,
  1641. and that has no place for the tfm widths. The PXL and PK
  1642. formats, on the other hand, contain the widths, so that if
  1643. those files are being used there is no need to open the TFM
  1644. file at all. Hence, this code should all be moved to after
  1645. we've figured out what kind of raster file is being used.]]
  1646. */ 
  1647.  
  1648. int define_font(e)
  1649. int e;
  1650. {
  1651.     int i;
  1652.     unsigned char p,n;
  1653.     if (e == MAXTEXFONTS) { fprintf(stderr,"\nToo many fonts for Ln03DVI\n");
  1654.     return(1); }
  1655.     txfa[nf] = (struct txf *)malloc(sizeof(struct txf));
  1656.     to_ext[nf] = e;
  1657.     four_bytes;
  1658.     four_bytes;
  1659.     txfa[nf] -> scaled_size = lcx.l;
  1660.     four_bytes;
  1661.     txfa[nf] -> design_size = lcx.l;
  1662.     for (i=0; i<256; i++) txfa[nf] -> chu[i] = '\0';
  1663.     p = mygetcdvi; n = mygetcdvi;
  1664.     font_name[nf] = malloc(p+n+1);
  1665.     for (i=0; i<p+n; i++) { font_name[nf][i] = mygetcdvi; }
  1666.     font_name[nf][p+n] = '\0';
  1667.  
  1668.     if (txfa[nf] -> scaled_size <= 0 || txfa[nf] -> scaled_size >=
  1669.     8*8*8*8*8*8*8*8*8) {
  1670.     fprintf(stderr,"\n Font %s has a bad scaled size\n",font_name[nf]);
  1671.     return(1);
  1672.     }
  1673.     if (txfa[nf] -> design_size <= 0 || txfa[nf] -> design_size >= 
  1674.     8*8*8*8*8*8*8*8*8) {
  1675.     fprintf(stderr,"\n Font %s has a bad design size\n",font_name[nf]);
  1676.     return(1);
  1677.     }
  1678.  
  1679. /* We follow DVItype and compute for each font a "space"
  1680. parameter which is one-sixth of the scaled design size. This
  1681. parameter is used in rounding the horizontal position to
  1682. pixels according to "Stanford rules." See the function set_h
  1683. below. */ 
  1684.  
  1685.     txfa[nf] -> space = txfa[nf] -> scaled_size/6;     
  1686.  
  1687.     i = read_tfm_file();
  1688.     if (i != 0) return(i);
  1689.     nf++;
  1690.     return(0); 
  1691.  
  1692. }
  1693.  
  1694. /* TFM files, like the DVI file, are read with getc, foolish
  1695. as that may seem. We use the same overlays that are used for
  1696. merging DVI bytes into longwords. However, since the TFM
  1697. file consists of longwords, only tfm_longword is need. */ 
  1698.  
  1699. #define tfm_longword { lcx.uc[3] = getc(tfmfile); \
  1700.     lcx.uc[2] = getc(tfmfile); lcx.uc[1] = getc(tfmfile); \
  1701.     lcx.uc[0] = getc(tfmfile); }
  1702.  
  1703. /* Read_tfm_file obtains the character widths from the TFM file
  1704. corresponding to font nf. */ 
  1705.  
  1706. int read_tfm_file() {
  1707.  
  1708.     int i,lh,nw;
  1709.     int info[256];
  1710.     long z,alpha,beta;
  1711.     long width[256];
  1712.  
  1713.     open_tfm_file();
  1714.     if (tfmfile == NULL) {
  1715.     fprintf(stderr,"\n Can't open TFM file for font ");
  1716.     perror(font_name[nf]);
  1717.     return(1); }
  1718.  
  1719. /* The TFM format is described in some issue of the TUGBoat
  1720. (TeX users' group newsletter), and in the comments to the
  1721. programs TeX and TFtoPL. Here we summarize those aspects of
  1722. TFM format that are relevant to the task of extracting the
  1723. widths of the characters. 
  1724.  
  1725. The first 24 bytes of a TFM file contain twelve 16-bit
  1726. integers that give the lengths of the various subsequent
  1727. portions of the file. The ones relevant to our purposes are
  1728. LH, length of the header data, BC, the smallest character
  1729. code in the font, EC, the largest character code in the
  1730. font, and NW, number of words in the width table. So, we
  1731. read those right now, and then skip over the remainder, and
  1732. then over a set of LH longwords called the "header." */ 
  1733.  
  1734.     tfm_longword; 
  1735.     lh = 256*lcx.uc[1] + lcx.uc[0];
  1736.     tfm_longword;
  1737.     txfa[nf] -> bc = 256*lcx.uc[3] + lcx.uc[2];
  1738.     txfa[nf] -> ec = 256*lcx.uc[1] + lcx.uc[0];
  1739.     tfm_longword;
  1740.     nw = 256*lcx.uc[3] + lcx.uc[2];
  1741.     if (txfa[nf] -> bc > 255 || txfa[nf] -> ec > 255 ||
  1742.     txfa[nf] -> bc > txfa[nf] -> ec || nw > 256) {
  1743.     fprintf(stderr,"\n Bad TFM file for font %s\n",font_name[nf]);
  1744.     return(1);
  1745.     }
  1746.  
  1747.     for(i=0; i<lh+3; i++) tfm_longword;
  1748.  
  1749. /* After the header, there are two arrays in the TFM file
  1750. that interest us. The first, INFO, is EC-BC+1 longwords long
  1751. and contains pointers to the second, WIDTH, which is NW
  1752. longwords long. For each character i, the width of i is
  1753. WIDTH[INFO[i-BC]]. We read these arrays into memory. */ 
  1754.  
  1755.     for(i=0; i<txfa[nf] -> ec-txfa[nf] -> bc+1; i++) {
  1756.     tfm_longword;
  1757.     if (lcx.uc[3] >= nw) {
  1758.         printf("\n Bad TFM file for font %s",font_name[nf]);
  1759.         return(1);
  1760.     }
  1761.     info[i] = lcx.uc[3];
  1762.     }
  1763.  
  1764. /* The widths are stored in a rather strange format known as
  1765. a "fix-word." A nonnegative width is expressed in "fix-word"
  1766. format by expressing it in units of 2^-20 times the design
  1767. size. A negative width is expressed in "fix-word" format by
  1768. expressing its negative in those units, and then changing
  1769. the most significant byte to 255. 
  1770.  
  1771. One needs to convert these widths into DVI units,
  1772. multiplying by the scaled design size according to a certain
  1773. arcane algorithm. The algorithm is copied from DVItype, to
  1774. which we refer the reader for an explanation. [[Copout. I
  1775. don't really understand what's going on here. Why not just
  1776. use floating point? That's what it's for.]] */ 
  1777.  
  1778.     z = txfa[nf] -> scaled_size;
  1779.     alpha = 16*z; beta = 16;
  1780.     while (z >= 4*(8*8*8*8*8*8*8)) { z = z/2; beta = beta/2; }
  1781.     for(i=0; i<nw; i++) {
  1782.     tfm_longword;
  1783.     width[i] = ( ( (lcx.uc[0]*z)/256 + lcx.uc[1]*z )/256  + 
  1784.         lcx.uc[2]*z)/beta;
  1785.     if (lcx.uc[3] == 255) width[i] -= alpha;
  1786.     }
  1787.  
  1788.     fclose(tfmfile);
  1789.  
  1790. /* Using these two arrays, we now compute the widths of the
  1791. characters in the fonts and place them into two malloc'ed
  1792. arrays, one for the widths in DVI units, one for the width
  1793. in pixel units. */ 
  1794.  
  1795.     font_width[nf] = (long int *)malloc(4*(txfa[nf] -> ec+1));
  1796.     
  1797.     for (i=0; i<txfa[nf] -> bc; i++) font_width[nf][i] = 0;
  1798.     for (i=txfa[nf] -> bc; i <= txfa[nf] -> ec; i++) 
  1799.     font_width[nf][i] = width[info[i-txfa[nf] -> bc]];
  1800.     return(0);
  1801. }
  1802.  
  1803. /* Open_tfm_file finds and opens the tfm file corresponding
  1804. to a font nf. The global file variable tfmfile is used to
  1805. hold the file pointer. [[This function will not work if the
  1806. TFM file lies over the net on a VMS V3.x host, because C
  1807. file opens do not work in such circumstances. The problem
  1808. can be ignored, because eventually there should be very few
  1809. VMS V3.x hosts left.]] */ 
  1810.  
  1811. int open_tfm_file () {
  1812.  
  1813.     int jext,jnam;
  1814.     char filespec[FILESPECLEN];
  1815.  
  1816. #ifndef vms
  1817.     char *texfontdir;
  1818. #endif
  1819.  
  1820.  
  1821.     find_filename(font_name[nf],&jnam,&jext);
  1822.     filespec[0] = '\0';
  1823.  
  1824. /* If there is no directory part, fill in using the logical
  1825. tex$fonts */ 
  1826.  
  1827. #ifdef vms
  1828.     if (jnam == 0) strcpy(filespec,"tex$fonts:");
  1829. #else
  1830.     if (jnam == 0) { 
  1831.     texfontdir = getenv("TEXFONTS");
  1832.     if  (texfontdir == NULL)
  1833.         texfontdir = "/usr/lib/tex/fonts";
  1834.     strcpy(filespec,texfontdir);
  1835.     if  (filespec[strlen(filespec)-1] != '/')
  1836.         strcat(filespec,"/");
  1837.     }
  1838. #endif
  1839.     strcat(filespec,font_name[nf]);
  1840.  
  1841. /* if there is no extension, add the extension ".tfm" */
  1842.  
  1843.     if (font_name[nf][jext] == '\0') strcat(filespec,".tfm");
  1844.     tfmfile = fopen(filespec,"r");
  1845.  
  1846. }
  1847.  
  1848. /* Define_font_pass2 skips a font definition from the DVI
  1849. file. */ 
  1850.  
  1851. int define_font_pass2()
  1852. {
  1853.     unsigned char p,n;
  1854.     int i;
  1855.     four_bytes;
  1856.     four_bytes;
  1857.     four_bytes;
  1858.     p = mygetcdvi; n = mygetcdvi;
  1859.     for (i=0; i<p+n; i++) mygetcdvi;
  1860. }
  1861.  
  1862. /* Do_rule writes into the ln3 file the escape sequence
  1863. corresponding to a rule. The rule is silently clipped to fit
  1864. on the page. */ 
  1865.  
  1866. int do_rule(xx0,yy0,xx1,yy1) 
  1867. int xx0,yy0,xx1,yy1;
  1868. {
  1869.     int j;
  1870.  
  1871.     xx0 = min(xx0+hoff,maxrmar);
  1872.     xx1 = min(xx1+hoff,maxrmar);
  1873.     xx0 = max(xx0,0);
  1874.     xx1 = max(xx1,0);
  1875.     yy0 = min(yy0+voff,maxbmar);
  1876.     yy1 = min(yy1+voff,maxbmar);
  1877.     yy0 = max(yy0,0);
  1878.     yy1 = max(yy1,0);
  1879.     if (xx0 > xx1) { j = xx0; xx0 = xx1; xx1 = j; }
  1880.     if (yy0 > yy1) { j = yy0; yy0 = yy1; yy1 = j; }
  1881.         
  1882.     if ((yy1 != yy0 && xx1 != xx0)) {
  1883.     fprintf(outfile,"\033[1;%d;%d;%d;%d!|",
  1884.         xx0,yy0,yy1-yy0,xx1-xx0);
  1885.     inc_ln3p(25);
  1886.     }
  1887. }
  1888.  
  1889. /* Set_v is called to execute vertical-position-altering
  1890. commands, other than pops. It modifies v and vv, and outputs
  1891. the vertical position to the LN3 file. We don't follow
  1892. Stanford rules here. They sometimes set the vertical
  1893. position in pixels vv to something other than the rounded
  1894. value of v. Not following Stanford rules implies, for
  1895. example, that the relative vertical positions of an accent
  1896. and its accentee may differ by one pixel according to how
  1897. the baseline of the accentee gets rounded. [[This should
  1898. perhaps be fixed.]] */ 
  1899.  
  1900. int set_v(v1)
  1901. int v1;
  1902. {
  1903.     int l;
  1904.  
  1905.     l = pixel_round(v1);
  1906.     v = v1;
  1907.     vv = l;
  1908.     vpset = 0;
  1909. }    
  1910.  
  1911. /* Set_h is called whenever a DVI command is encountered
  1912. that alters the horizonal position, other than a set_char,
  1913. set_rule or pop. It sets h to the new value, and alters hh
  1914. according to Stanford rules. */ 
  1915.  
  1916. int set_h(new_h)
  1917. int new_h;   
  1918. {
  1919.     int l,old_hh;
  1920.  
  1921.     old_hh = hh;
  1922.     l = pixel_round(new_h);
  1923.     if (txfa[curf] == 0 || new_h-h >= txfa[curf] -> space
  1924.     || new_h-h <= -4*txfa[curf] -> space) hh = l;
  1925.     else {
  1926.     hh += pixel_round(new_h-h);
  1927.     if (hh-l > MAX_DRIFT) hh = l+MAX_DRIFT;
  1928.     else if (hh-l < -MAX_DRIFT) hh = l-MAX_DRIFT;
  1929.     }
  1930.     h = new_h;
  1931.     return(0);
  1932. }
  1933.